﻿// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License. See LICENSE in the project root for license information.

using Microsoft.UI.Xaml.Tests.MUXControls.ApiTests.RepeaterTests.Common;
using Microsoft.UI.Xaml.Tests.MUXControls.ApiTests.RepeaterTests.Common.Mocks;
using MUXControlsTestApp.Utilities;
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.Threading;
using Windows.Foundation;
using Microsoft.UI.Xaml.Controls;
using Microsoft.UI.Xaml.Controls.Primitives;
using Microsoft.UI.Xaml.Markup;
using Common;

using WEX.TestExecution;
using WEX.TestExecution.Markup;
using WEX.Logging.Interop;

namespace Microsoft.UI.Xaml.Tests.MUXControls.ApiTests.RepeaterTests
{
    [TestClass]
    public class FlowLayoutTests : ApiTestBase
    {
        [TestMethod]
        public void ValidateStackLayout()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int panelMinorSize = 400;
                    const int numItems = 5;
                    const int itemMajorSize = 50;
                    // We test for when the item is smaller than the panel and when it's bigger than the panel: 
                    int[] itemMinorSizeChoices = new int[] { 200, 800 };

                    foreach (int itemMinorSize in itemMinorSizeChoices)
                    {

                        LayoutPanel panel = new LayoutPanel()
                        {
                            Layout = new StackLayout() { Orientation = scrollOrientation.ToLayoutOrientation() }
                        };

                        SetPanelMinorSize(panel, om, panelMinorSize);
                        for (int i = 0; i < numItems; i++)
                        {
                            panel.Children.Add(
                                new Button()
                                {
                                    Content = i,
                                    Width = om.IsVertical ? itemMinorSize : itemMajorSize,
                                    Height = om.IsVertical ? itemMajorSize : itemMinorSize
                                });
                        }

                        Content = panel;
                        Content.UpdateLayout();
                        int itemSpacing = 0;
                        ValidateStackLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], panelMinorSize, numItems, itemMajorSize, itemSpacing, panel.DesiredSize);

                        itemSpacing = 10;
                        ((StackLayout)panel.Layout).Spacing = itemSpacing;
                        panel.UpdateLayout();
                        ValidateStackLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], panelMinorSize, numItems, itemMajorSize, itemSpacing, panel.DesiredSize);
                    }
                }
            });
        }

        [TestMethod]
        public void ValidateGridLayout()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int panelMinorSize = 400;
                    const int numItems = 10;
                    const int itemMajorSize = 50;
                    int[] itemMinorSizeChoices = new int[] { 100, 800 };

                    foreach (int itemMinorSize in itemMinorSizeChoices)
                    {
                        Log.Comment(string.Format("MinorSize: {0}", itemMinorSize));
                        LayoutPanel panel = new LayoutPanel()
                        {
                            Layout = new UniformGridLayout()
                            {
                                Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                                ItemsJustification = UniformGridLayoutItemsJustification.Start,
                                MinItemWidth = om.IsVertical ? itemMinorSize : itemMajorSize,
                                MinItemHeight = om.IsVertical ? itemMajorSize : itemMinorSize
                            },
                            VerticalAlignment = VerticalAlignment.Top,
                            HorizontalAlignment = HorizontalAlignment.Left
                        };

                        SetPanelMinorSize(panel, om, panelMinorSize);
                        for (int i = 0; i < numItems; i++)
                        {
                            panel.Children.Add(new Button() { Content = i });
                        }

                        // Put the panel inside a canvas to give it all the space it needs
                        // and avoid any clipping of desired size by the framework.
                        var canvas = new Canvas();
                        canvas.Children.Add(panel);
                        Content = canvas;
                        Content.UpdateLayout();
                        int minRowSpacing = 0;
                        int minColumnSpacing = 0;
                        ValidateGridLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], itemMinorSize, itemMajorSize, minRowSpacing, minColumnSpacing, panel.Children.Count, panel.DesiredSize);

                        var layout = (UniformGridLayout)panel.Layout;
                        minRowSpacing = 5;
                        minColumnSpacing = 10;
                        layout.MinRowSpacing = minRowSpacing;
                        layout.MinColumnSpacing = minColumnSpacing;
                        Content.UpdateLayout();
                        ValidateGridLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], itemMinorSize, itemMajorSize, minRowSpacing, minColumnSpacing, panel.Children.Count, panel.DesiredSize);

                        // Change the properties and validate that they are reflected in the 
                        // layout bounds of children once layout has completed.
                        minRowSpacing = 5;
                        minColumnSpacing = 10;
                        layout.MinRowSpacing = minRowSpacing;
                        layout.MinColumnSpacing = minColumnSpacing;
                        layout.MinItemWidth *= 2;
                        layout.MinItemHeight *= 2;
                        Content.UpdateLayout();
                        ValidateGridLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], itemMinorSize*2, itemMajorSize*2, minRowSpacing, minColumnSpacing, panel.Children.Count, panel.DesiredSize);
                    }
                }
            });
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateResizingFirstItemResizesOtherItemsInGridLayout()
        {
            RunOnUIThread.Execute(() =>
            {
                const int numItems = 10;
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    foreach (DimensionChoice dimension in Enum.GetValues(typeof(DimensionChoice)))
                    {
                        LayoutPanel panel = new LayoutPanel()
                        {
                            Layout = new UniformGridLayout() { }
                        };

                        for (int i = 0; i < numItems; i++)
                        {
                            panel.Children.Add(new Button() { Content = i, HorizontalAlignment = HorizontalAlignment.Stretch });
                        }

                        Content = panel;
                        Content.UpdateLayout();

                        // verify the defaults before our modification 
                        for (int i = 0; i < panel.Children.Count; i++)
                        {
                            var child = (FrameworkElement)panel.Children.ElementAt(i);
                            var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                            // Ensure width is not 100 before updating.
                            Verify.AreNotEqual(100, layoutBounds.Width);
                            Verify.AreEqual(30, layoutBounds.Height);
                        }

                        // modify the first item's width, height, or both
                        var firstItem = panel.Children[0] as Button;

                        switch (dimension)
                        {
                            case DimensionChoice.Width:
                                firstItem.Width = 100;
                                break;
                            case DimensionChoice.Height:
                                firstItem.Height = 100;
                                break;
                            case DimensionChoice.Size:
                                firstItem.Width = 100;
                                firstItem.Height = 100;
                                break;
                            default:
                                throw new InvalidOperationException("Unhandled dimension choice.");
                        }

                        // Need to invalidate measure to kick in bigger size, 
                        // If the new size is larger, then setting the value does not
                        // invalidate measure, because the desired size is going to be same.
                        panel.InvalidateMeasure();
                        Content.UpdateLayout();

                        // validate that our dimension's size has propagated to all other buttons
                        for (int i = 0; i < panel.Children.Count; i++)
                        {
                            var child = (FrameworkElement)panel.Children.ElementAt(i);
                            var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                            switch (dimension)
                            {
                                case DimensionChoice.Width:
                                    Verify.AreEqual(100, layoutBounds.Width);
                                    break;
                                case DimensionChoice.Height:
                                    Verify.AreEqual(100, layoutBounds.Height);
                                    break;
                                case DimensionChoice.Size:
                                    Verify.AreEqual(100, layoutBounds.Width);
                                    Verify.AreEqual(100, layoutBounds.Height);
                                    break;
                                default:
                                    throw new InvalidOperationException("Unhandled dimension choice.");

                            }
                        }

                        // Make the first item smaller
                        switch (dimension)
                        {
                            case DimensionChoice.Width:
                                firstItem.Width = 32;
                                break;
                            case DimensionChoice.Height:
                                firstItem.Height = 32;
                                break;
                            case DimensionChoice.Size:
                                firstItem.Width = 32;
                                firstItem.Height = 32;
                                break;
                            default:
                                throw new InvalidOperationException("Unhandled dimension choice.");
                        }

                        Content.UpdateLayout();

                        // validate that our dimension's size has propagated to all other buttons
                        for (int i = 0; i < panel.Children.Count; i++)
                        {
                            var child = (FrameworkElement)panel.Children.ElementAt(i);
                            var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                            switch (dimension)
                            {
                                case DimensionChoice.Width:
                                    Verify.AreEqual(32, layoutBounds.Width);
                                    break;
                                case DimensionChoice.Height:
                                    Verify.AreEqual(32, layoutBounds.Height);
                                    break;
                                case DimensionChoice.Size:
                                    Verify.AreEqual(32, layoutBounds.Width);
                                    Verify.AreEqual(32, layoutBounds.Height);
                                    break;
                                default:
                                    throw new InvalidOperationException("Unhandled dimension choice.");

                            }
                        }
                    }
                }
            });
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateDefaultWidthForGridLayoutItemsIsBasedOnFirstItem()
        {
            ValidateDimensionForGridLayoutItemsIsBasedOnFirstItem(DimensionChoice.Width);
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateDefaultHeightForGridLayoutItemsIsBasedOnFirstItem()
        {
            ValidateDimensionForGridLayoutItemsIsBasedOnFirstItem(DimensionChoice.Height);
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateDefaultSizeForGridLayoutItemsIsBasedOnFirstItem()
        {
            ValidateDimensionForGridLayoutItemsIsBasedOnFirstItem(DimensionChoice.Size);
        }

        [TestMethod]
        public void ValidateDefaultSizeForFirstItemDoesntOverridePresetOne()
        {
            RunOnUIThread.Execute(() =>
            {
                const int itemWidth = 50;
                const int itemHeight = 50;

                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new UniformGridLayout()
                        {
                            MinItemWidth = itemWidth,
                            MinItemHeight = itemHeight,
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                        }
                    };

                    PopulatePanelWithButtons(panel, 10 /*numItems*/);
                    Content = panel;
                    Content.UpdateLayout();

                    ModifyElementSize(panel, 0 /*index*/, 100 /*itemWidth*/, 100 /*itemHeight*/);
                    Content = panel;
                    Content.UpdateLayout();
                    ValidateElementsSizeInGridLayout(panel, itemWidth, itemHeight);
                }
            });
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateAdaptabilityWhenChangingFirstElementWidthForGridLayout()
        {
            ValidateAdaptabilityWhenChangingTheFirstElementForGridLayout(DimensionChoice.Width);
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateAdaptabilityWhenChangingFirstElementHeightForGridLayout()
        {
            ValidateAdaptabilityWhenChangingTheFirstElementForGridLayout(DimensionChoice.Height);
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateAdaptabilityWhenChangingFirstElementSizeForGridLayout()
        {
            ValidateAdaptabilityWhenChangingTheFirstElementForGridLayout(DimensionChoice.Size);
        }

        [TestMethod]
        public void ValidateGridLayoutJustificationAndStretch()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation, useLayoutRounding: true);
                    const int panelMinorSize = 500;
                    const int numItems = 2;
                    const int itemMinorSize = 200;
                    const int itemMajorSize = 100;
                    Log.Comment("UniformGridLayoutItemsJustification.Start");
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new UniformGridLayout()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                            ItemsJustification = UniformGridLayoutItemsJustification.Start,
                            MinItemWidth = om.IsVertical ? itemMinorSize : itemMajorSize,
                            MinItemHeight = om.IsVertical ? itemMajorSize : itemMinorSize
                        }
                    };

                    SetPanelMinorSize(panel, om, panelMinorSize);
                    for (int i = 0; i < numItems; i++)
                    {
                        panel.Children.Add(new Button() { Content = i });
                    }

                    Content = panel;
                    Content.UpdateLayout();
                    var layout = (UniformGridLayout)panel.Layout;

                    Log.Comment("UniformGridLayoutItemsJustification.Center");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Center;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(50, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(250, 0, itemMinorSize, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.End");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.End;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(100, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(300, 0, itemMinorSize, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.SpaceBetween");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.SpaceBetween;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(300, 0, itemMinorSize, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.SpaceAround");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.SpaceAround;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(25, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(275, 0, itemMinorSize, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.SpaceEvenly");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.SpaceEvenly;

                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(33.3333, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(266.6667, 0, itemMinorSize, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.Start +  UniformGridLayoutItemsStretch.Fill");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Start;
                    layout.ItemsStretch = UniformGridLayoutItemsStretch.Fill;
                    layout.MinRowSpacing = 10;
                    layout.MinColumnSpacing = 10;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, itemMinorSize + 45, itemMajorSize),
                            om.MinorMajorRect(255, 0, itemMinorSize + 45, itemMajorSize)
                        });

                    Log.Comment("UniformGridLayoutItemsJustification.Start +  UniformGridLayoutItemsStretch.Uniform");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Start;
                    layout.ItemsStretch = UniformGridLayoutItemsStretch.Uniform;
                    layout.MinRowSpacing = 10;
                    layout.MinColumnSpacing = 10;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, itemMinorSize + 45, itemMajorSize + 22.5),
                            om.MinorMajorRect(255, 0, itemMinorSize + 45, itemMajorSize + 22.5)
                        });
                }
            });
        }

        [TestMethod]
        public void ValidateGridLayoutMaximumRowsOrColumns()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation, useLayoutRounding: true);
                    const int panelMinorSize = 500;
                    const int numItems = 4;
                    const int itemMinorSize = 150;
                    const int itemMajorSize = 100;
                    Log.Comment("UniformGridLayoutItemsJustification.Start + UniformGridLayoutItemsStretch.None");
                    LayoutPanel panel = new LayoutPanel();
                    var layout = new UniformGridLayout()
                    {
                        Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                        ItemsJustification = UniformGridLayoutItemsJustification.Start,
                        MinItemWidth = om.IsVertical ? itemMinorSize : itemMajorSize,
                        MinItemHeight = om.IsVertical ? itemMajorSize : itemMinorSize
                    };

                    SetPanelMinorSize(panel, om, panelMinorSize);
                    for (int i = 0; i < numItems; i++)
                    {
                        panel.Children.Add(new Button() { Content = i });
                    }

                    panel.Layout = layout;
                    Content = panel;
                    panel.UpdateLayout();

                    Log.Comment($"MaximumRowsOrColumns = {layout.MaximumRowsOrColumns}");
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(150, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(300, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(0, 100, itemMinorSize, itemMajorSize)
                        });

                    layout.MaximumRowsOrColumns = 0;
                    Log.Comment($"MaximumRowsOrColumns = {layout.MaximumRowsOrColumns}");
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(0, 100, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(0, 200, itemMinorSize, itemMajorSize),
                            om.MinorMajorRect(0, 300, itemMinorSize, itemMajorSize)
                        });

                    layout.MaximumRowsOrColumns = 1;
                    Log.Comment($"MaximumRowsOrColumns = {layout.MaximumRowsOrColumns} + UniformGridLayoutItemsJustification.Start +  UniformGridLayoutItemsStretch.Uniform");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Start;
                    layout.ItemsStretch = UniformGridLayoutItemsStretch.Uniform;
                    layout.MinRowSpacing = 10;
                    layout.MinColumnSpacing = 10;
                    var aspectRatio = itemMinorSize / (double)itemMajorSize;
                    var extraMajorSize = (panelMinorSize - itemMinorSize) / aspectRatio;
                    var majorOffset = itemMajorSize + extraMajorSize + layout.MinRowSpacing;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0,               panelMinorSize, itemMajorSize + extraMajorSize),
                            om.MinorMajorRect(0, majorOffset,     panelMinorSize, itemMajorSize + extraMajorSize),
                            om.MinorMajorRect(0, majorOffset * 2, panelMinorSize, itemMajorSize + extraMajorSize),
                            om.MinorMajorRect(0, majorOffset * 3, panelMinorSize, itemMajorSize + extraMajorSize),
                        });

                    layout.MaximumRowsOrColumns = 2;
                    Log.Comment($"MaximumRowsOrColumns = {layout.MaximumRowsOrColumns} + UniformGridLayoutItemsJustification.Start +  UniformGridLayoutItemsStretch.Fill");
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Start;
                    layout.ItemsStretch = UniformGridLayoutItemsStretch.Fill;
                    layout.MinRowSpacing = 10;
                    layout.MinColumnSpacing = 10;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0,    0,                  itemMinorSize + 95, itemMajorSize),
                            om.MinorMajorRect(255,  0,                  itemMinorSize + 95, itemMajorSize),
                            om.MinorMajorRect(0,    itemMajorSize + 10, itemMinorSize + 95, itemMajorSize),
                            om.MinorMajorRect(255,  itemMajorSize + 10, itemMinorSize + 95, itemMajorSize),
                        });

                }
            });
        }

        [TestMethod]
        public void ValidateGridLayoutDesiredSizingWithMaxRowsOrColumnsSet()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation, useLayoutRounding: true);
                    const int numItems = 4;

                    int itemSize = 100;

                    LayoutPanel panel = new LayoutPanel();
                    var layout = new UniformGridLayout() {
                        Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                        ItemsJustification = UniformGridLayoutItemsJustification.Start,
                        MinItemWidth = itemSize,
                        MinItemHeight = itemSize,
                        MinRowSpacing = 0,
                        MinColumnSpacing = 0
                    };
                    
                    for (int i = 0; i < numItems; i++)
                    {
                        panel.Children.Add(new Button() { Content = i,Width = itemSize,Height = itemSize });
                    }
                    
                    panel.Layout = layout;
                    Content = panel;

                    // Place in top left corner to prevent stretch of panel
                    panel.VerticalAlignment = VerticalAlignment.Top;
                    panel.HorizontalAlignment = HorizontalAlignment.Left;
                    layout.ItemsJustification = UniformGridLayoutItemsJustification.Start;

                    for(int i = 1; i < numItems; i++)
                    {
                        layout.MaximumRowsOrColumns = i;
                        panel.UpdateLayout();

                        if (om.IsVertical)
                        {
                            // Vertical so fill columns first
                            Verify.AreEqual(itemSize * i, panel.DesiredSize.Width);
                            Verify.AreEqual(Math.Ceiling((double)numItems / (double)i) * itemSize, panel.DesiredSize.Height);
                        }
                        else
                        {
                            // Horizontal, fill rows first
                            Verify.AreEqual(itemSize * i, panel.DesiredSize.Height);
                            Verify.AreEqual(Math.Ceiling((double)numItems / (double)i) * itemSize, panel.DesiredSize.Width);
                        }
                    }

                }
            });
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621.
        public void ValidateFlowLayout()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    int[] panelMinorSizeChoices = new int[] { 10, 50 };
                    foreach (int panelMinorSize in panelMinorSizeChoices)
                    {
                        const int numItems = 10;
                        LayoutPanel panel = new LayoutPanel()
                        {
                            Layout = new FlowLayout()
                            {
                                Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                                LineAlignment = FlowLayoutLineAlignment.Start
                            }
                        };

                        SetPanelMinorSize(panel, om, panelMinorSize);
                        for (int i = 0; i < numItems; i++)
                        {
                            panel.Children.Add(new Button() { Content = i });
                        }

                        Content = panel;

                        Content.UpdateLayout();
                        int minItemSpacing = 0;
                        int lineSpacing = 0;
                        ValidateFlowLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], minItemSpacing, lineSpacing, panel.Children.Count, panel.DesiredSize);

                        minItemSpacing = 10;
                        lineSpacing = 10;
                        ((FlowLayout)panel.Layout).MinRowSpacing = minItemSpacing;
                        ((FlowLayout)panel.Layout).MinColumnSpacing = lineSpacing;
                        Content.UpdateLayout();
                        ValidateFlowLayoutChildrenLayoutBounds(om, (i) => panel.Children[i], minItemSpacing, lineSpacing, panel.Children.Count, panel.DesiredSize);
                    }
                }
            });
        }

        [TestMethod]
        public void ValidateFlowLayoutVaryingHeights()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int numItems = 10;
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new FlowLayout()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                            LineAlignment = FlowLayoutLineAlignment.Start
                        }
                    };

                    SetPanelMinorSize(panel, om, 180);
                    for (int i = 0; i < numItems; i++)
                    {
                        var child = new Button() { Content = i };
                        if (scrollOrientation == ScrollOrientation.Vertical)
                        {
                            child.Width = 50;
                            child.Height = 50 + i % 2 * 50;
                        }
                        else
                        {
                            child.Width = 50 + i % 2 * 50;
                            child.Height = 50;
                        }

                        panel.Children.Add(child);
                    }

                    Content = panel;

                    Content.UpdateLayout();
                    int minItemSpacing = 0;
                    int lineSpacing = 0;
                    Log.Comment("Validate with no spacing");
                    ValidateFlowLayoutChildrenLayoutBounds(
                        om,
                        (i) => panel.Children[i],
                        minItemSpacing,
                        lineSpacing,
                        panel.Children.Count,
                        panel.DesiredSize,
                        scrollOrientation == ScrollOrientation.Vertical ? 50 : 100,
                        scrollOrientation == ScrollOrientation.Vertical ? 100 : 50);

                    minItemSpacing = 10;
                    lineSpacing = 10;
                    ((FlowLayout)panel.Layout).MinRowSpacing = minItemSpacing;
                    ((FlowLayout)panel.Layout).MinColumnSpacing = lineSpacing;
                    Content.UpdateLayout();
                    Log.Comment("Validate with spacing");
                    ValidateFlowLayoutChildrenLayoutBounds(
                        om,
                        (i) => panel.Children[i],
                        minItemSpacing,
                        lineSpacing,
                        panel.Children.Count,
                        panel.DesiredSize,
                        scrollOrientation == ScrollOrientation.Vertical ? 50 : 100,
                        scrollOrientation == ScrollOrientation.Vertical ? 100 : 50);
                }
            });
        }

        [TestMethod]
        public void ValidateFlowLayoutLineAlignment()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int panelMinorSize = 500;
                    const int itemMajorSize = 100;
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new FlowLayout()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                            LineAlignment = FlowLayoutLineAlignment.Start
                        }
                    };

                    SetPanelMinorSize(panel, om, panelMinorSize);
                    panel.Children.Add(new Button()
                    {
                        Content = 0,
                        Width = om.IsVertical ? 100 : itemMajorSize,
                        Height = om.IsVertical ? itemMajorSize : 100,
                    });

                    panel.Children.Add(new Button()
                    {
                        Content = 1,
                        Width = om.IsVertical ? 200 : itemMajorSize,
                        Height = om.IsVertical ? itemMajorSize : 200,
                    });

                    Content = panel;
                    Content.UpdateLayout();
                    var layout = (FlowLayout)panel.Layout;
                    layout.LineAlignment = FlowLayoutLineAlignment.Center;
                    panel.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(100, 0, 100, itemMajorSize),
                            om.MinorMajorRect(200, 0, 200, itemMajorSize)
                        });

                    layout.LineAlignment = FlowLayoutLineAlignment.End;
                    Content.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(200, 0, 100, itemMajorSize),
                            om.MinorMajorRect(300, 0, 200, itemMajorSize)
                        });

                    layout.LineAlignment = FlowLayoutLineAlignment.SpaceBetween;
                    Content.UpdateLayout();
                    ValidateChildBounds(
                        panel,
                        new List<Rect>()
                        {
                            om.MinorMajorRect(0, 0, 100, itemMajorSize),
                            om.MinorMajorRect(300, 0, 200, itemMajorSize)
                        });
                }
            });
        }

        [TestMethod]
        public void ValidateMakeAnchor()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("Orientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);

                    const int numItems = 1000;
                    int getElementCallCount = 0;
                    int recycleElementCallCount = 0;
                    var elementFactory = new RecyclingElementFactoryDerived()
                    {
                        Templates = { { "key", GetDataTemplate("<Button Content='{Binding}' />") } },
                        RecyclePool = new RecyclePool(),
                        GetElementFunc = (int index, UIElement owner, UIElement elementFromBase) =>
                        {
                            getElementCallCount++;
                            return elementFromBase;
                        },
                        ClearElementFunc = (UIElement element, UIElement owner) => recycleElementCallCount++,
                    };

                    ItemsRepeater repeater = null;
                    ScrollViewer scrollViewer = null;
                    Content = CreateAndInitializeRepeater
                    (
                       om,
                       itemsSource: Enumerable.Range(0, numItems),
                       elementFactory: elementFactory,
                       layout: new FlowLayoutDerived()
                       {
                           Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                       },
                       repeater: ref repeater,
                       scrollViewer: ref scrollViewer
                    );

                    Content.UpdateLayout();
                    // Validate that we are indeed virtualizing
                    Verify.IsGreaterThan(numItems, elementFactory.RealizedElementIndices.Count);

                    // GetAnchor on an element in view.
                    getElementCallCount = 0;
                    var anchor = repeater.GetOrCreateElement(1);
                    Verify.AreEqual(1, repeater.GetElementIndex(anchor));
                    Verify.AreEqual(0, getElementCallCount);

                    // GetAnchor on element outside the view.
                    recycleElementCallCount = 0;
                    int realizedElementsBeforeMakeAnchor = elementFactory.RealizedElementIndices.Count();
                    anchor = repeater.GetOrCreateElement(900);
                    Verify.AreEqual(900, repeater.GetElementIndex(anchor));
                    Verify.AreEqual(1, getElementCallCount);

                    Content.UpdateLayout();
                    Verify.AreEqual(realizedElementsBeforeMakeAnchor, recycleElementCallCount);
                    Verify.IsLessThan(1, elementFactory.RealizedElementIndices.Count());
                    Verify.IsGreaterThan(150, elementFactory.RealizedElementIndices.Count());

                    Verify.Throws<ArgumentException>(() => repeater.GetOrCreateElement(numItems + 10));
                }
            });
        }

        // TODO: Bug 11136935 - Add test coverage for GetAnchorElement once we have a programatic way to 
        // provide the suggested anchor.
        // Tests should include 
        //- connected, 
        //- disconnected
        //- suggest an anchor that is valid
        //- suggest an anchor that is not valid(some random uielement)
        //- suggest an anchor that on gridlayout that gives a different target index.
        //- !valid anchor index
        //  - change size, itemspacing - getanchorforrealizationrect
        //  - window not connected  - anchorforvisiblerect
        //  - connected window - first in realized range should be picked.

        // Validate that the extent and realized items look fine with connected and disconnected
        // jumps for stack/grid and flow layouts.

        [TestMethod]
        public void ValidateEstimations_Stack_Horizontal()
        {
            ValidateLayoutEstimations(ScrollOrientation.Horizontal, LayoutChoice.Stack);
        }

        [TestMethod]
        [TestProperty("TestPass:MaxOSVer", "22621")] // This test is currently failing on 23h2, hence stop at 22h2 which is 22621
        public void ValidateEstimations_Stack_Vertical()
        {
            ValidateLayoutEstimations(ScrollOrientation.Vertical, LayoutChoice.Stack);
        }

        [TestMethod]
        public void ValidateEstimations_Grid_Horizontal()
        {
            ValidateLayoutEstimations(ScrollOrientation.Horizontal, LayoutChoice.Grid);
        }

        [TestMethod]
        [TestProperty("IsolationLevel", "Method")] // Task 28232821: DCPP Test: ItemsRepeater tests are running in isolation mode due to test instability.
        public void ValidateEstimations_Grid_Vertical()
        {
            ValidateLayoutEstimations(ScrollOrientation.Vertical, LayoutChoice.Grid);
        }

        [TestMethod]
        public void ValidateEstimations_Flow_Horizontal()
        {
            ValidateLayoutEstimations(ScrollOrientation.Horizontal, LayoutChoice.Flow);
        }

        [TestMethod]
        public void ValidateEstimations_Flow_Vertical()
        {
            ValidateLayoutEstimations(ScrollOrientation.Vertical, LayoutChoice.Flow);
        }

        [TestMethod]
        public void VerifyFlowLayoutOnLineArranged()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("Orientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int panelMinorSize = 100;
                    const double itemSize = 50;
                    const int numItems = 5;
                    int onLineArrangedCallCount = 0;
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new FlowLayoutDerived()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                            OnLineArrangedFunc = (int startIndex, int countInLine, double lineSize, VirtualizingLayoutContext context) =>
                            {
                                Verify.AreEqual(0, startIndex % 2);
                                Verify.AreEqual(lineSize, itemSize);
                                // Last line has only one item in it
                                Verify.AreEqual(startIndex == 4 ? 1 : 2, countInLine);
                                onLineArrangedCallCount++;
                            }
                        }
                    };

                    SetPanelMinorSize(panel, om, panelMinorSize);
                    for (int i = 0; i < numItems; i++)
                    {
                        panel.Children.Add(new Button() { Content = i, Width = itemSize, Height = itemSize });
                    }

                    Content = panel;

                    Content.UpdateLayout();

                    Verify.AreEqual(3, onLineArrangedCallCount);
                }
            });
        }

        [TestMethod]
        [TestProperty("IsolationLevel", "Method")] // Task 28232821: DCPP Test: ItemsRepeater tests are running in isolation mode due to test instability.
        public void ValidateLayoutSharing()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment(string.Format("Orientation: {0}", scrollOrientation));
                    var om = new OrientationBasedMeasures(scrollOrientation);
                    const int itemSize = 20;
                    var layouts = new List<VirtualizingLayout>()
                    {
                        new StackLayout(),
                        new UniformGridLayout(){ MinItemWidth = itemSize, MinItemHeight = itemSize },
                        new FlowLayoutDerived()
                    };

                    for (int layoutIndex = 0; layoutIndex < layouts.Count; layoutIndex++)
                    {
                        var repeater1Layout = layouts[layoutIndex];
                        var repeater2Layout = layouts[(layoutIndex + 1) % layouts.Count];
                        Log.Comment(string.Format(" Layouts: {0} and {1}", repeater1Layout.GetType(), repeater2Layout.GetType()));
                        repeater1Layout.SetOrientation(scrollOrientation);
                        repeater2Layout.SetOrientation(scrollOrientation);

                        const int repeater1NumItems = 5;
                        const int repeater2NumItems = 10;
                        var elementFactory = new RecyclingElementFactoryDerived()
                        {
                            Templates = { { "key", GetDataTemplate(@"<Button Content='{Binding}' Width='20' Height='20'/>") } },
                            RecyclePool = new RecyclePool(),
                            ValidateElementIndices = false, // can get same indicies for different repeaters
                        };

                        var repeater1 = new ItemsRepeater()
                        {
                            ItemsSource = Enumerable.Range(0, repeater1NumItems),
                            Layout = repeater1Layout,
                            ItemTemplate = elementFactory,
                            HorizontalCacheLength = 0,
                            VerticalCacheLength = 0,
                        };

                        var repeater2 = new ItemsRepeater()
                        {
                            ItemsSource = Enumerable.Range(0, repeater2NumItems),
                            Layout = repeater2Layout,
                            ItemTemplate = elementFactory,
                            HorizontalCacheLength = 0,
                            VerticalCacheLength = 0,
                        };

                        var stack = new StackPanel() { Orientation = scrollOrientation.ToLayoutOrientation() };
                        stack.Children.Add(repeater1);
                        stack.Children.Add(repeater2);

                        var scrollViewer = new ScrollViewer()
                        {
                            Content = stack
                        };

                        SetUpScrollViewerOrientation(scrollViewer, scrollOrientation);
                        Content = new ItemsRepeaterScrollHost()
                        {
                            Width = 400,
                            Height = 400,
                            ScrollViewer = scrollViewer
                        };

                        Content.UpdateLayout();

                        if (repeater1Layout is StackLayout)
                        {
                            ValidateStackLayoutChildrenLayoutBounds(om, (i) => repeater1.TryGetElement(i), 400, repeater1NumItems, itemSize, 0, repeater1.DesiredSize);
                        }
                        else if (repeater1Layout is UniformGridLayout)
                        {
                            ValidateGridLayoutChildrenLayoutBounds(om, (i) => repeater1.TryGetElement(i), itemSize, itemSize, 0, 0, repeater1NumItems, repeater1.DesiredSize);
                        }
                        else
                        {
                            ValidateFlowLayoutChildrenLayoutBounds(om, (i) => repeater1.TryGetElement(i), 0, 0, repeater1NumItems, repeater1.DesiredSize);
                        }

                        if (repeater2Layout is StackLayout)
                        {
                            ValidateStackLayoutChildrenLayoutBounds(om, (i) => repeater2.TryGetElement(i), 400, repeater2NumItems, itemSize, 0, repeater2.DesiredSize);
                        }
                        else if (repeater2Layout is UniformGridLayout)
                        {
                            ValidateGridLayoutChildrenLayoutBounds(om, (i) => repeater2.TryGetElement(i), itemSize, itemSize, 0, 0, repeater2NumItems, repeater2.DesiredSize);
                        }
                        else
                        {
                            ValidateFlowLayoutChildrenLayoutBounds(om, (i) => repeater2.TryGetElement(i), 0, 0, repeater2NumItems, repeater2.DesiredSize);
                        }
                    }
                }
            });
        }

        [TestMethod]
        [TestProperty("Bug", "12042780")]
        public void ValidateLayoutWithInfiniteMeasureSizeInNonVirtualizingDirection()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    Log.Comment("ScrollOrientation is " + scrollOrientation);

                    var data = new ObservableCollection<int>(Enumerable.Range(0, 4));
                    var dataSource = MockItemsSource.CreateDataSource(data, supportsUniqueIds: false);
                    var elementFactory = MockElementFactory.CreateElementFactory(
                        Enumerable.Range(0, data.Count).Select(i => new ContentControl
                        {
                            Width = (i + 1) * 40,
                            Height = (i + 1) * 40,
                        }).ToList());

                    var repeater = new ItemsRepeater();
                    repeater.ItemsSource = dataSource;
                    repeater.ItemTemplate = elementFactory;

                    var scrollViewer = new ScrollViewer();
                    scrollViewer.HorizontalScrollMode = ScrollMode.Enabled;
                    scrollViewer.HorizontalScrollBarVisibility = ScrollBarVisibility.Visible;
                    scrollViewer.Content = repeater;
                    scrollViewer.Width = scrollViewer.Height = 600;

                    var tracker = new ItemsRepeaterScrollHost();
                    tracker.ScrollViewer = scrollViewer;

                    Content = tracker;
                    Content.UpdateLayout();

                    var layouts = new VirtualizingLayout[]
                    {
                        new StackLayout { Spacing = 10, Orientation = scrollOrientation.ToLayoutOrientation() },
                        new UniformGridLayout { MinItemHeight = 40, MinItemWidth = 40, MinColumnSpacing = 10, MinRowSpacing = 10, Orientation = scrollOrientation.ToOrthogonalLayoutOrientation() },
                        new FlowLayout { MinColumnSpacing = 10, MinRowSpacing = 10, Orientation = scrollOrientation.ToOrthogonalLayoutOrientation() }
                    };

                    var verifyDesiredSize = new Action<Size, Size>((expected, actual) =>
                    {
                        Verify.AreEqual(expected.Width, scrollOrientation == ScrollOrientation.Vertical ? actual.Width : actual.Height);
                        Verify.AreEqual(expected.Height, scrollOrientation == ScrollOrientation.Vertical ? actual.Height : actual.Width);
                    });

                    foreach (var layout in layouts)
                    {
                        Log.Comment("Layout is " + layout.GetType().Name);
                        repeater.Layout = layout;
                        Content.UpdateLayout();

                        if (layout is StackLayout) verifyDesiredSize(new Size(160, 430), repeater.DesiredSize);
                        if (layout is UniformGridLayout) verifyDesiredSize(new Size(190, 40), repeater.DesiredSize);
                        if (layout is FlowLayout) verifyDesiredSize(new Size(430, 160), repeater.DesiredSize);
                    }
                }
            });
        }

        // Bug 17411124: StackLayout: measuring multiple times causes extent to sometimes be 0 in the non-virtualizing direction.
        [TestMethod]
        [TestProperty("Bug", "12052938")]
        public void ValidateIntersectionWithRealizationWindow()
        {
            // In this test, we have a ScrollViewer with a viewport of 200x200.
            // The content is a stack panel with a 300x300 border followed by a ItemsRepeater.
            // At the beginning, the ItemsRepeater doesn't intersect the visible or realization window.
            // The test will wait for the realization window to expand and will validate that we
            // realize elements when it begins to intersect the repeater.

            var createLayouts = new Func<ScrollOrientation, VirtualizingLayout[]>((scrollOrientation) =>
            {
                VirtualizingLayout[] layouts = null;
                RunOnUIThread.Execute(() =>
                {
                    layouts = new VirtualizingLayout[]
                    {
                        new StackLayout { Spacing = 10, Orientation = scrollOrientation.ToLayoutOrientation() },
                        new UniformGridLayout { MinItemHeight = 40, MinItemWidth = 40, MinColumnSpacing = 10, MinRowSpacing = 10, Orientation = scrollOrientation.ToOrthogonalLayoutOrientation() },
                        new FlowLayout { MinColumnSpacing = 10, MinRowSpacing = 10, Orientation = scrollOrientation.ToOrthogonalLayoutOrientation() }
                    };
                });
                return layouts;
            });

            foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
            {
                foreach (var layout in createLayouts(scrollOrientation))
                {
                    Log.Comment(" Layout: " + layout.GetType().Name + " ScrollOrientation:" + scrollOrientation.ToString());
                    var repeater = (ItemsRepeater)null;
                    var rangeRealizedEvent = new AutoResetEvent(initialState: false);
                    var data = new ObservableCollection<string>(Enumerable.Range(0, 4).Select(i => "Item #" + i));
                    var mapping = (List<ContentControl>)null;

                    RunOnUIThread.Execute(() =>
                    {
                        mapping = Enumerable.Range(0, data.Count).Select(i => new ContentControl { Width = 40, Height = 40 }).ToList();

                        var dataSource = MockItemsSource.CreateDataSource(data, supportsUniqueIds: false);
                        var elementFactory = MockElementFactory.CreateElementFactory(mapping);

                        repeater = new ItemsRepeater();
                        // Explicitly set a cache length so we dont depend on the defaults.
                        repeater.VerticalCacheLength = 3;
                        repeater.HorizontalCacheLength = 3;
                        repeater.ItemsSource = dataSource;
                        repeater.ItemTemplate = elementFactory;
                        repeater.Layout = layout;

                        var stack = new StackPanel();
                        stack.Orientation = scrollOrientation.ToLayoutOrientation();
                        stack.Children.Add(new Border { Width = 300, Height = 300 });
                        stack.Children.Add(repeater);

                        var scrollViewer = new ScrollViewer();
                        scrollViewer.Content = stack;
                        scrollViewer.Width = scrollViewer.Height = 200;
                        SetUpScrollViewerOrientation(scrollViewer, scrollOrientation);

                        var tracker = new ItemsRepeaterScrollHost();
                        tracker.ScrollViewer = scrollViewer;
                        Content = tracker;

                        repeater.ElementPrepared += (o, e) =>
                        {
                            if (e.Index == data.Count - 1)
                            {
                                for (int i = 0; i < data.Count; ++i)
                                {
                                    Verify.AreEqual(mapping[i], repeater.TryGetElement(i));
                                }

                                rangeRealizedEvent.Set();
                            }
                        };

                        Content.UpdateLayout();
                    });

                    Verify.IsTrue(rangeRealizedEvent.WaitOne(DefaultWaitTimeInMS), "Waiting for all items to be realized as the realization window expands.");
                }
            }
        }

        [TestMethod]
        public void ValidateUniformGridWithNoItems()
        {
            RunOnUIThread.Execute(() =>
            {
                var repeater = new ItemsRepeater() {
                    ItemsSource = new List<string>(), // no items 
                    Layout = new UniformGridLayout() { Orientation = Orientation.Vertical },
                };

                Content = repeater;
                Content.UpdateLayout();

                // Ensure we do not crash and get zero size.
                Verify.AreEqual(0, repeater.DesiredSize.Width);
                Verify.AreEqual(0, repeater.DesiredSize.Height);
            });
        }

        [TestMethod]
        public void ValidateUniformGridWithItemsRelyingOnAspectRatio()
        {
            ItemsRepeater repeater = null;
            ScrollViewer scroller = null;
            RunOnUIThread.Execute(() =>
            {
                repeater = new ItemsRepeater() {
                    ItemsSource = Enumerable.Range(0, 6), // no items 
                    Layout = new UniformGridLayout() {
                        Orientation = Orientation.Vertical,
                        MaximumRowsOrColumns = 1,
                        MinItemWidth = 100
                    },
                    MaxWidth = 300,
                    ItemTemplate = GetDataTemplate("<repeaterCommon:AspectRatioRespectingControl/>")
                };

                scroller = new ScrollViewer() {
                    Content = repeater
                };

                Content = scroller;
                repeater.UpdateLayout();
                Content.UpdateLayout();
            });

            IdleSynchronizer.Wait();

            RunOnUIThread.Execute(() =>
            {
                // Assert not implemented yet
                Verify.IsTrue(Math.Abs(300 - repeater.DesiredSize.Width) < 1, "Repeater width should be 300");
                Verify.IsTrue(repeater.DesiredSize.Height > 1, "Repeater should have height");
            });
        }

        [TestMethod]
        public void ValidateFlowLayoutWithOneItemHasNonZeroExtent()
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    var repeater = new ItemsRepeater()
                    {
                        ItemsSource = new List<string>() { "single item" },
                        Layout = new FlowLayout() { Orientation = scrollOrientation.ToLayoutOrientation() },
                    };

                    Content = repeater;
                    Content.UpdateLayout();

                    Verify.IsTrue(repeater.ActualWidth > 10);
                    Verify.IsTrue(repeater.ActualHeight > 10);
                }
            });
        }

        // Make sure that GridLayout does not end up de-virtualizing if scrolling in the 
        // same orientation as items are being laid out. It will layout in one line and 
        // never wrap, but we should avoid devirtualizing.
        [TestMethod]
        public void ValidateGridLayoutWithSameOrientationAsScrolling()
        {
            ManualResetEvent viewChanged = new ManualResetEvent(false);
            ScrollViewer scrollViewer = null;
            var lastRealizedIndex = int.MinValue;
            var firstRealizedIndex = int.MaxValue;
            RunOnUIThread.Execute(() =>
            {

                var repeater = new ItemsRepeater()
                {
                    ItemsSource = Enumerable.Range(1, 1000),
                    ItemTemplate = GetDataTemplate(@"<Button Content='{Binding}' Width='100' Height='100'/>"),
                    Layout = new UniformGridLayout() { Orientation = Orientation.Vertical },
                    HorizontalCacheLength = 0,
                    VerticalCacheLength = 0,
                };

                repeater.ElementPrepared += (sender, args) =>
                {
                    firstRealizedIndex = Math.Min(firstRealizedIndex, args.Index);
                    lastRealizedIndex = Math.Max(lastRealizedIndex, args.Index);
                };

                scrollViewer = new ScrollViewer()
                {
                    Content = repeater,
                    Height = 200,
                };

                var anchorProvier = new ItemsRepeaterScrollHost()
                {
                    ScrollViewer = scrollViewer
                };

                Content = anchorProvier;
                Content.UpdateLayout();

                Verify.IsLessThan(lastRealizedIndex - firstRealizedIndex, 10);

                scrollViewer.ViewChanged += (sender, args) =>
                {
                    viewChanged.Set();
                };

            });

            for (int i = 0; i < 3; i++)
            {
                viewChanged.Reset();
                RunOnUIThread.Execute(() =>
                {
                    lastRealizedIndex = int.MinValue;
                    firstRealizedIndex = int.MaxValue;
                    scrollViewer.ChangeView(horizontalOffset: 0, verticalOffset: (i + 1) * 80, zoomFactor: null, disableAnimation: true);
                });

                IdleSynchronizer.Wait();
                Verify.IsTrue(viewChanged.WaitOne(DefaultWaitTime));

                RunOnUIThread.Execute(() =>
                {
                    // we used to crash due to a layout cycle before we get here.
                    Verify.IsLessThan(lastRealizedIndex - firstRealizedIndex, 10);
                });
            }
        }

        // Make sure the UniformGridLayout handles infinite available measure size
        // when items are sized and ItemsStretch is other than None.
        [TestMethod]
        public void VerifyUnconstrainedUniformGridLayout()
        {
            RunOnUIThread.Execute(() =>
            {
                var scrollViewer = XamlReader.Load(
                    @"<ScrollViewer xmlns='http://schemas.microsoft.com/winfx/2006/xaml/presentation'
                        HorizontalScrollBarVisibility='Auto'>
                        <LayoutPanel>
                            <LayoutPanel.Layout>
                                <UniformGridLayout ItemsStretch='Fill' MinItemWidth='100' MinItemHeight='100'/>
                            </LayoutPanel.Layout>
                            <Border/>
                        </LayoutPanel>
                    </ScrollViewer>") as ScrollViewer;

                Content = scrollViewer;
                scrollViewer.UpdateLayout();
            });
            IdleSynchronizer.Wait();
        }

        [TestMethod]
        public void VerifyItemsGetFullSpaceInMajorDirectionWhenSmallerThanLineSize()
        {
            RunOnUIThread.Execute(() =>
            {
                var layoutPanel = (LayoutPanel)XamlReader.Load(
                    @"<controls:LayoutPanel Width='800'  
                        xmlns='http://schemas.microsoft.com/winfx/2006/xaml/presentation'
                        xmlns:controls='using:Microsoft.UI.Xaml.Controls'>
                        <controls:LayoutPanel.Layout>
                            <controls:FlowLayout  />
                        </controls:LayoutPanel.Layout>
                        <Border Width='300' Height='300' />
                        <Border Width='350' Height='150' />
                        <Border Width='300' Height='300' />
                        <Border Width='350' Height='250' />
                    </controls:LayoutPanel>");

                Content = layoutPanel;
                layoutPanel.UpdateLayout();

                var expected = new List<Rect>() 
                {
                    new Rect(0, 0, 300, 300),
                    new Rect(300, 0, 350, 300),
                    new Rect(0, 300, 300, 300),
                    new Rect(300, 300, 350, 300),
                    new Rect(0, 600, 200, 250)
                };

                for (int i = 0; i < layoutPanel.Children.Count; i++)
                {
                    var child = (FrameworkElement)layoutPanel.Children[i];
                    var actualRect = LayoutInformation.GetLayoutSlot(child);
                    Verify.AreEqual(expected[i], actualRect);
                }
            });
        }

        [TestMethod]
        public void ValidateStackLayoutArrangeWithStretch()
        {
            RunOnUIThread.Execute(() =>
            {
                var svWidth = 1000.0;
                ScrollViewer sv = new ScrollViewer() {
                    Width = svWidth,
                    HorizontalScrollBarVisibility = ScrollBarVisibility.Auto
                };

                var repeater = new ItemsRepeater();
                repeater.ItemsSource = Enumerable.Range(0, 10);
                repeater.ItemTemplate = GetDataTemplate(@"<TextBlock HorizontalAlignment='Stretch' Text='{Binding}' />");

                sv.Content = repeater;

                Content = sv;
                Content.UpdateLayout();

                var firstItem = (FrameworkElement)repeater.TryGetElement(0);
                var bounds = LayoutInformation.GetLayoutSlot(firstItem);
                Verify.AreEqual(svWidth, bounds.Width);

            });
        }

        [TestMethod]
        public void ValidateStackLayoutArrangeWithInsertion()
        {
            RunOnUIThread.Execute(() =>
            {
                var itemsSource = new ObservableCollection<int>(Enumerable.Range(0, 3));
                var sp = new StackPanel();
                var sl = new StackLayout()
                {
                    Orientation = Orientation.Horizontal,
                    Spacing = 0
                };
                var repeater = new ItemsRepeater();
                repeater.Layout = sl;
                repeater.ItemsSource = itemsSource;
                repeater.ItemTemplate = GetDataTemplate(@"<TextBlock Width='64' Height='32' Text='{Binding}'/>");

                sp.Children.Add(repeater);

                Content = sp;
                Content.UpdateLayout();

                Log.Comment("Item bounds before insertion:");

                for (int index = 0; index < 3; index++)
                {
                    var item = repeater.TryGetElement(index) as FrameworkElement;
                    var actualBounds = LayoutInformation.GetLayoutSlot(item);
                    var expectedBounds = new Rect(64 * index, 0, 64, 32);

                    Log.Comment(string.Format(@"  Index:{0}, Expected:{1} Actual:{2}", index, expectedBounds, actualBounds));
                    Verify.AreEqual(expectedBounds, actualBounds);
                }

                for (int insertionIndex = 0; insertionIndex < 3; insertionIndex++)
                {
                    Log.Comment(string.Format(@"Item {0} insertion at index:{1}", itemsSource.Count, insertionIndex));
                    itemsSource.Insert(insertionIndex, itemsSource.Count);

                    Content.UpdateLayout();

                    Log.Comment("Item bounds after insertion:");

                    for (int index = 0; index < itemsSource.Count; index++)
                    {
                        var item = repeater.TryGetElement(index) as FrameworkElement;
                        var actualBounds = LayoutInformation.GetLayoutSlot(item);
                        var expectedBounds = new Rect(64 * index, 0, 64, 32);

                        Log.Comment(string.Format(@"  Index:{0}, Expected:{1} Actual:{2}", index, expectedBounds, actualBounds));
                        Verify.AreEqual(expectedBounds, actualBounds);
                    }
                }
            });
        }

        #region Private Helpers

        private enum LayoutChoice
        {
            Stack,
            Grid,
            Flow
        }

        private enum DimensionChoice
        {
            Width,
            Height,
            Size
        }

        private void ValidateStackLayoutChildrenLayoutBounds(
            OrientationBasedMeasures om,
            Func<int, UIElement> elementAtIndexFunc,
            int panelMinorSize,
            int numItems,
            int itemMajorSize,
            int itemSpacing,
            Size desiredSize)
        {
            Rect expectedRect = om.MinorMajorRect(0, 0, panelMinorSize, itemMajorSize);
            for (int i = 0; i < numItems; i++)
            {
                var child = (FrameworkElement)elementAtIndexFunc(i);
                var actualRect = LayoutInformation.GetLayoutSlot(child);
                Verify.AreEqual(expectedRect, actualRect);

                om.SetMajorStart(ref expectedRect, om.MajorStart(expectedRect) + itemMajorSize + itemSpacing);
            }

            Verify.AreEqual(numItems * itemMajorSize + (numItems - 1) * itemSpacing, om.Major(desiredSize));
        }

        private void ValidateGridLayoutChildrenLayoutBounds(
            OrientationBasedMeasures om,
            Func<int, UIElement> elementAtIndexFunc,
            double itemMinorSize,
            double itemMajorSize,
            double minRowSpacing,
            double minColumnSpacing,
            int childCount,
            Size desiredSize)
        {
            var expectedRect = om.MinorMajorRect(0, 0, itemMinorSize, itemMajorSize);
            double extentMajor = 0;
            var minItemSpacing = om.ScrollOrientation == ScrollOrientation.Vertical ? minColumnSpacing : minRowSpacing;
            var lineSpacing = om.ScrollOrientation == ScrollOrientation.Vertical ? minRowSpacing : minColumnSpacing;

            for (int i = 0; i < childCount; i++)
            {
                var child = (FrameworkElement)elementAtIndexFunc(i);
                var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                Verify.AreEqual(expectedRect, layoutBounds);

                extentMajor = om.MajorStart(expectedRect) + itemMajorSize;
                om.SetMinorStart(ref expectedRect, om.MinorStart(expectedRect) + itemMinorSize + minItemSpacing);
                if (om.MinorStart(expectedRect) + itemMinorSize + minItemSpacing > om.Minor(desiredSize))
                {
                    om.SetMinorStart(ref expectedRect, 0);
                    om.SetMajorStart(ref expectedRect, om.MajorStart(expectedRect) + itemMajorSize + lineSpacing);
                }
            }

            Verify.AreEqual(extentMajor, om.Major(desiredSize));
        }

        private void ValidateFlowLayoutChildrenLayoutBounds(
            OrientationBasedMeasures om,
            Func<int, UIElement> elementAtIndexFunc,
            double minItemSpacing,
            double lineSpacing,
            int childCount,
            Size desiredSize,
            float? expectedItemWidth = null,
            float? expectedItemHeight = null)
        {
            var expectedRect = om.MinorMajorRect(0, 0, 0, 0);
            double extentMajor = 0;
            double lineSize = 0;
            Log.Comment("ValidateFlowLayoutChildrenLayoutBounds");
            for (int i = 0; i < childCount; i++)
            {
                var child = (FrameworkElement)elementAtIndexFunc(i);

                var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                expectedRect.Width = expectedItemWidth.HasValue ? expectedItemWidth.Value : child.DesiredSize.Width;
                expectedRect.Height = expectedItemHeight.HasValue ? expectedItemHeight.Value : child.DesiredSize.Height;
                lineSize = Math.Max(lineSize, om.Major(child.DesiredSize));

                Log.Comment(string.Format(@"Index:{0}, Expected:{1} Actual:{2}", i, expectedRect, layoutBounds));
                Verify.AreEqual(expectedRect, layoutBounds);

                extentMajor = om.MajorStart(expectedRect) + lineSize;
                om.SetMinorStart(ref expectedRect, om.MinorStart(expectedRect) + om.Minor(child.DesiredSize) + minItemSpacing);
                if (om.MinorStart(expectedRect) + om.Minor(child.DesiredSize) + minItemSpacing > om.Minor(desiredSize))
                {
                    om.SetMinorStart(ref expectedRect, 0);
                    om.SetMajorStart(ref expectedRect, om.MajorStart(expectedRect) + lineSize + lineSpacing);
                }
            }

            Log.Comment(string.Format(@"Extent Expected:{0} Actual:{1}", om.Major(desiredSize), extentMajor));
            Verify.AreEqual(extentMajor, om.Major(desiredSize));
        }

        private void ValidateChildBounds(LayoutPanel panel, List<Rect> list)
        {
            for (int i = 0; i < list.Count; i++)
            {
                var child = (FrameworkElement)panel.Children[i];
                Verify.AreEqual(list[i], LayoutInformation.GetLayoutSlot(child));
            }
        }

        private DataTemplate GetDataTemplate(string content)
        {
            return (DataTemplate)XamlReader.Load(
                       string.Format(@"<DataTemplate  
                            xmlns='http://schemas.microsoft.com/winfx/2006/xaml/presentation'
                            xmlns:repeaterCommon='using:MUXControls.ApiTests.RepeaterTests.Common'>
                           {0}
                        </DataTemplate>", content));
        }

        private ItemsRepeaterScrollHost CreateAndInitializeRepeater(
           OrientationBasedMeasures om,
           object itemsSource,
           VirtualizingLayout layout,
           ElementFactory elementFactory,
           ref ItemsRepeater repeater,
           ref ScrollViewer scrollViewer)
        {
            repeater = new ItemsRepeater()
            {
                ItemsSource = itemsSource,
                Layout = layout,
                ItemTemplate = elementFactory,
                HorizontalCacheLength = 0,
                VerticalCacheLength = 0,
            };

            scrollViewer = new ScrollViewer()
            {
                Content = repeater
            };

            SetUpScrollViewerOrientation(scrollViewer, om.ScrollOrientation);
            return new ItemsRepeaterScrollHost()
            {
                Width = 400,
                Height = 400,
                ScrollViewer = scrollViewer
            };
        }

        private void ValidateLayoutEstimations(ScrollOrientation scrollOrientation, LayoutChoice layoutChoice)
        {
            Log.Comment(string.Format("ScrollOrientation: {0}", scrollOrientation));
            var om = new OrientationBasedMeasures(scrollOrientation);
            List<VirtualizingLayout> layouts = new List<VirtualizingLayout>();
            const double itemSize = 90;
            const double lineSpacing = 10;
            double itemSizeWithLineSpacing = itemSize + lineSpacing;
            VirtualizingLayout layout = null;
            var viewChangedEvent = new ManualResetEvent(false);

            RunOnUIThread.Execute(() =>
            {
                switch (layoutChoice)
                {
                    case LayoutChoice.Stack:
                        layout = new StackLayout() { Spacing = lineSpacing };
                        break;

                    case LayoutChoice.Grid:
                        {
                            var minRowSpacing = om.ScrollOrientation == ScrollOrientation.Vertical ? lineSpacing : 0;
                            var minColumnSpacing = om.ScrollOrientation == ScrollOrientation.Horizontal ? lineSpacing : 0;
                            layout = new UniformGridLayout() { MinItemWidth = itemSize, MinItemHeight = itemSize, MinRowSpacing = minRowSpacing, MinColumnSpacing = minColumnSpacing };
                        }
                        break;

                    case LayoutChoice.Flow:
                        {
                            var minRowSpacing = om.ScrollOrientation == ScrollOrientation.Vertical ? lineSpacing : 0;
                            var minColumnSpacing = om.ScrollOrientation == ScrollOrientation.Horizontal ? lineSpacing : 0;
                            layout = new FlowLayoutDerived()
                            {
                                MinRowSpacing = minRowSpacing,
                                MinColumnSpacing = minColumnSpacing,
                                OnLineArrangedFunc = (int startIndex, int countInLine, double lineSize, VirtualizingLayoutContext context) =>
                                {
                                    Verify.AreEqual(0, startIndex % 4);
                                    Verify.AreEqual(4, countInLine);
                                    Verify.AreEqual(lineSize, itemSize);
                                }
                            };
                        }
                        break;

                    default:
                        throw new InvalidOperationException("Unhandled layout choice.");
                }
            });

            ItemsRepeater repeater = null;
            RecyclingElementFactoryDerived elementFactory = null;
            ScrollViewer scrollViewer = null;
            const int numItems = 1000;
            var numColumns = (int)(400 / itemSize);
            var numRows = numItems / numColumns;

            RunOnUIThread.Execute(() =>
            {
                Log.Comment(string.Format(" Layout: {0}", layout.GetType()));
                layout.SetOrientation(scrollOrientation);

                elementFactory = new RecyclingElementFactoryDerived()
                {
                    Templates = { { "key", GetDataTemplate(@"<Button Content='{Binding}' Width='90' Height='90'/>") } },
                    RecyclePool = new RecyclePool(),
                };

                Content = CreateAndInitializeRepeater
                (
                   om,
                   itemsSource: Enumerable.Range(0, numItems),
                   elementFactory: elementFactory,
                   layout: layout,
                   repeater: ref repeater,
                   scrollViewer: ref scrollViewer
                );

                scrollViewer.ViewChanged += (sender, args) =>
                {
                    if (!args.IsIntermediate)
                    {
                        viewChangedEvent.Set();
                    }
                };

                Content.UpdateLayout();
                if (layout is StackLayout)
                {
                    Verify.AreEqual(numItems * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is UniformGridLayout || layout is FlowLayout)
                {
                    Verify.AreEqual(numRows * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
            });

            // Move window - connected
            IdleSynchronizer.Wait();

            RunOnUIThread.Execute(() =>
            {
                var success = om.IsVertical ?
                    scrollViewer.ChangeView(null, 200, null, true) :
                    scrollViewer.ChangeView(200, null, null, true);
            });

            Verify.IsTrue(viewChangedEvent.WaitOne(DefaultWaitTime), "Waiting for ViewChanged.");
            IdleSynchronizer.Wait();

            RunOnUIThread.Execute(() =>
            {
                if (layout is StackLayout)
                {
                    Verify.AreEqual(numItems * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is UniformGridLayout || layout is FlowLayout)
                {
                    Verify.AreEqual(numRows * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
            });

            // Move window - disconnected
            RunOnUIThread.Execute(() =>
            {
                viewChangedEvent.Reset();
                var success = om.IsVertical ?
                   scrollViewer.ChangeView(null, 20000, null, true) :
                   scrollViewer.ChangeView(20000, null, null, true);
            });

            Verify.IsTrue(viewChangedEvent.WaitOne(DefaultWaitTime), "Waiting for ViewChanged.");
            IdleSynchronizer.Wait();

            RunOnUIThread.Execute(() =>
            {
                if (layout is StackLayout)
                {
                    VerifyRealizedRange(expectedStartindex: 199, expectedCount: 6, actualIndices: elementFactory.RealizedElementIndices);
                    Verify.AreEqual(numItems * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is UniformGridLayout)
                {
                    // We have to copy the list we're passing, as any modification in the realizedElementIndices will result in 
                    // altering the behavior and preventing the recycling of the containers.
                    List<int> observedIndices = new List<int>(elementFactory.RealizedElementIndices);
                    observedIndices.Remove(0);
                    VerifyRealizedRange(expectedStartindex: 799, expectedCount: 18, actualIndices: observedIndices);
                    Verify.AreEqual(numRows * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is FlowLayout)
                {
                    VerifyRealizedRange(expectedStartindex: 799, expectedCount: 18, actualIndices: elementFactory.RealizedElementIndices);
                    Verify.AreEqual(numRows * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }

            });

            // Validate Ability to override Extent using a disconnected move
            RunOnUIThread.Execute(() =>
            {
                if (layout is FlowLayoutDerived)
                {
                    var flowLayout = (FlowLayoutDerived)layout;
                    flowLayout.GetExtentFunc = (
                       Size availableSize,
                       VirtualizingLayoutContext context,
                       UIElement firstRealized,
                       int firstRealizedItemIndex,
                       Rect firstRealizedLayoutBounds,
                       UIElement lastRealized,
                       int lastRealizedItemIndex,
                       Rect lastRealizedLayoutBounds,
                       Rect fromBase) =>
                    {
                        return om.MinorMajorRect(0, 0, 400, numRows * itemSize);
                    };
                }

                viewChangedEvent.Reset();
                var success = om.IsVertical ?
                   scrollViewer.ChangeView(null, 10000, null, true) :
                   scrollViewer.ChangeView(10000, null, null, true);
            });

            Verify.IsTrue(viewChangedEvent.WaitOne(DefaultWaitTime), "Waiting for ViewChanged.");
            IdleSynchronizer.Wait();

            RunOnUIThread.Execute(() =>
            {
                if (layout is StackLayout)
                {
                    Verify.AreEqual(numItems * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is UniformGridLayout)
                {
                    Verify.AreEqual(numRows * itemSizeWithLineSpacing - lineSpacing, om.Major(repeater.DesiredSize));
                }
                else if (layout is FlowLayout)
                {
                    // Extent overriden using GetExtent override.
                    Verify.AreEqual(numRows * itemSize, om.Major(repeater.DesiredSize));
                }
            });
        }

        private void ValidateDimensionForGridLayoutItemsIsBasedOnFirstItem(DimensionChoice dimension)
        {
            LayoutPanel panel = null;
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    panel = new LayoutPanel()
                    {
                        Layout = new UniformGridLayout()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                        }
                    };

                    PopulatePanelWithButtons(panel, 10 /*numItems*/);
                    Content = panel;
                    Content.UpdateLayout();

                    double widthBeforeUpdate = double.NaN;
                    for (int i = 0; i < panel.Children.Count; i++)
                    {
                        var child = (FrameworkElement)panel.Children.ElementAt(i);
                        var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                        // Ensure bounds before updating.
                        Verify.AreNotEqual(100, layoutBounds.Width);
                        widthBeforeUpdate = layoutBounds.Width;
                        Verify.AreEqual(30 /* itemHeight */, layoutBounds.Height);
                    }

                    switch (dimension)
                    {
                        case DimensionChoice.Width:
                            ModifyElementSize(panel, 0 /*index*/, 100 /*itemWidth*/, Double.NaN /*itemHeight*/);
                            Content.InvalidateMeasure(); // Bug 16644056
                            Content.UpdateLayout();
                            ValidateElementsSizeInGridLayout(panel, 100 /*itemWidth*/, 30 /*itemHeight*/);
                            break;

                        case DimensionChoice.Height:

                            ModifyElementSize(panel, 0 /*index*/, Double.NaN /*itemWidth*/, 100 /*itemHeight*/);
                            Content.InvalidateMeasure(); // Bug 16644056
                            Content.UpdateLayout();
                            ValidateElementsSizeInGridLayout(panel, widthBeforeUpdate /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        case DimensionChoice.Size:
                            ModifyElementSize(panel, 0 /*index*/, 100 /*itemWidth*/, 100 /*itemHeight*/);
                            Content.InvalidateMeasure(); // Bug 16644056
                            Content.UpdateLayout();
                            ValidateElementsSizeInGridLayout(panel, 100 /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        default:
                            throw new InvalidOperationException("Unhandled dimension choice.");
                    }
                }
            });
        }

        private void ValidateAdaptabilityWhenChangingTheFirstElementForGridLayout(DimensionChoice dimension)
        {
            RunOnUIThread.Execute(() =>
            {
                foreach (ScrollOrientation scrollOrientation in Enum.GetValues(typeof(ScrollOrientation)))
                {
                    LayoutPanel panel = new LayoutPanel()
                    {
                        Layout = new UniformGridLayout()
                        {
                            Orientation = scrollOrientation.ToOrthogonalLayoutOrientation(),
                        }
                    };

                    PopulatePanelWithButtons(panel, 10);
                    Content = panel;
                    Content.UpdateLayout();

                    double widthBeforeUpdate = double.NaN;
                    for (int i = 0; i < panel.Children.Count; i++)
                    {
                        var child = (FrameworkElement)panel.Children.ElementAt(i);
                        var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                        // Ensure bounds before updating.
                        Verify.AreNotEqual(100, layoutBounds.Width);
                        widthBeforeUpdate = layoutBounds.Width;
                        Verify.AreEqual(30 /* itemHeight */, layoutBounds.Height);
                    }

                    switch (dimension)
                    {
                        case DimensionChoice.Width:
                            ModifyElementSize(panel, 1 /*index*/, 100 /*itemWidth*/, Double.NaN /*itemHeight*/);
                            break;

                        case DimensionChoice.Height:

                            ModifyElementSize(panel, 1 /*index*/, Double.NaN /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        case DimensionChoice.Size:
                            ModifyElementSize(panel, 1 /*index*/, 100 /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        default:
                            throw new InvalidOperationException("Unhandled dimension choice.");
                    }

                    panel.Children.RemoveAt(0);
                    Content.UpdateLayout();

                    switch (dimension)
                    {
                        case DimensionChoice.Width:
                            ValidateElementsSizeInGridLayout(panel, 100 /*itemWidth*/, 30 /*itemHeight*/);
                            break;

                        case DimensionChoice.Height:
                            ValidateElementsSizeInGridLayout(panel, widthBeforeUpdate /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        case DimensionChoice.Size:
                            ValidateElementsSizeInGridLayout(panel, 100 /*itemWidth*/, 100 /*itemHeight*/);
                            break;

                        default:
                            throw new InvalidOperationException("Unhandled dimension choice.");
                    }

                }
            });
        }

        private static void VerifyWithinTolerance(double expected, double actual, double tolerance)
        {
            Log.Comment(string.Format("Verify: WithinTolerance({0}, {1}, {2})", expected, actual, tolerance));
            if (Math.Abs(expected - actual) > tolerance)
            {
                Verify.Fail(string.Format("AreEqual failed to be within tolerance: {0}:{1}:{2}", expected, actual, tolerance));
            }
        }

        private static void SetUpScrollViewerOrientation(ScrollViewer scrollViewer, ScrollOrientation scrollOrientation)
        {
            if (scrollOrientation == ScrollOrientation.Vertical)
            {
                scrollViewer.SetValue(ScrollViewer.HorizontalScrollModeProperty, ScrollMode.Disabled);
                scrollViewer.SetValue(ScrollViewer.VerticalScrollModeProperty, ScrollMode.Enabled);
                scrollViewer.SetValue(ScrollViewer.HorizontalScrollBarVisibilityProperty, ScrollBarVisibility.Disabled);
                scrollViewer.SetValue(ScrollViewer.VerticalScrollBarVisibilityProperty, ScrollBarVisibility.Visible);
            }
            else
            {
                scrollViewer.SetValue(ScrollViewer.HorizontalScrollModeProperty, ScrollMode.Enabled);
                scrollViewer.SetValue(ScrollViewer.VerticalScrollModeProperty, ScrollMode.Disabled);
                scrollViewer.SetValue(ScrollViewer.HorizontalScrollBarVisibilityProperty, ScrollBarVisibility.Visible);
                scrollViewer.SetValue(ScrollViewer.VerticalScrollBarVisibilityProperty, ScrollBarVisibility.Disabled);
            }
        }

        private static void VerifyRealizedRange(int expectedStartindex, int expectedCount, List<int> actualIndices)
        {
            Verify.AreEqual(expectedCount, actualIndices.Count);
            for (int i = expectedStartindex; i < expectedStartindex + expectedCount; i++)
            {
                Verify.IsTrue(actualIndices.Contains(i));
            }
        }


        private void SetPanelMinorSize(LayoutPanel panel, OrientationBasedMeasures om, double value)
        {
            if (om.IsVertical)
            {
                panel.Width = value;
            }
            else
            {
                panel.Height = value;
            }

        }

        private void PopulatePanelWithButtons(LayoutPanel panel, int numItems)
        {
            for (int i = 0; i < numItems; i++)
            {
                panel.Children.Add(new Button() { Content = i.ToString() });
            }
        }

        private static void ValidateElementsSizeInGridLayout(LayoutPanel panel, double itemWidth, double itemHeight)
        {
            for (int i = 0; i < panel.Children.Count; i++)
            {
                var child = (FrameworkElement)panel.Children.ElementAt(i);
                var layoutBounds = LayoutInformation.GetLayoutSlot(child);

                Verify.AreEqual(itemWidth, layoutBounds.Width);
                Verify.AreEqual(itemHeight, layoutBounds.Height);
            }
        }

        private static void ModifyElementSize(LayoutPanel panel, int index, double itemWidth, double itemHeight)
        {
            var item = (FrameworkElement)panel.Children.ElementAt(index);

            if (!Double.IsNaN(itemWidth))
            {
                item.Width = itemWidth;
            }
            if (!Double.IsNaN(itemHeight))
            {
                item.Height = itemHeight;
            }
        }

        private int DefaultWaitTime = 2000;

        #endregion
    }
}
